/*
 * SpiderWebChart.java
 * Android-Charts
 *
 * Created by limc on 2011/05/29.
 *
 * Copyright 2011 limc.cn All rights reserved.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

/**
 *
 */
package cn.limc.ohoscharts.view;


import cn.limc.ohoscharts.entity.TitleValueEntity;
import ohos.agp.components.AttrSet;
import ohos.agp.components.Component;
import ohos.agp.render.Canvas;
import ohos.agp.render.Paint;
import ohos.agp.render.Path;
import ohos.agp.utils.Color;
import ohos.agp.utils.Point;
import ohos.app.Context;

import java.util.ArrayList;
import java.util.List;

/**
 *
 * <p>
 * SpiderWebChart is a kind of graph that display data on web-like graph. each
 * data was displayed in the longitude lines,like a area graph.
 * </p>
 * <p>
 * SpiderWebChartは円グラフとアリアグラフを合わせるのグラフ一種です、データをワッブで表示します。
 * </p>
 * <p>
 * SpiderWebChart是一种将数据内容显示在网状图上的一种图表、是将饼图和面积图结合表示的一种图表
 * </p>
 *
 * @author limc
 * @version v1.0 2011/05/30 14:01:54
 *
 */
public class SpiderWebChart extends RoundChart implements Component.DrawTask {


  /**
   * <p>
   * default longitude numbers
   * </p>
   * <p>
   * 経線の数量のデフォルト値
   * </p>
   * <p>
   * 默认经线数量
   * </p>
   */
  public static final int DEFAULT_LONGITUDE_NUM = 5;

  /**
   * <p>
   * default should display latitude lines
   * </p>
   * <p>
   * 緯線を表示する
   * </p>
   * <p>
   * 默认是否显示纬线
   * </p>
   */
  public static final boolean DEFAULT_DISPLAY_LATITUDE = true;

  /**
   * <p>
   * default latitude numbers
   * </p>
   * <p>
   * 緯線の数量のデフォルト値
   * </p>
   * <p>
   * 默认纬线数量
   * </p>
   */
  public static final int DEFAULT_LATITUDE_NUM = 5;

  /**
   * <p>
   * default color for latitude lines
   * </p>
   * <p>
   * 緯線の色のデフォルト値
   * </p>
   * <p>
   * 默认纬线颜色
   * </p>
   */
  public static final int DEFAULT_LATITUDE_COLOR = Color.BLACK.getValue();

  /**
   * <p>
   * default color for display data
   * </p>
   * <p>
   * データの色のデフォルト値
   * </p>
   * <p>
   * 默认数据的显示颜色
   * </p>
   */
  public static final int[] COLORS = {Color.RED.getValue(), Color.BLUE.getValue(),
      Color.YELLOW.getValue()};

  /**
   * <p>
   * Data
   * </p>
   * <p>
   * データ
   * </p>
   * <p>
   * 图表数据
   * </p>
   */
  protected List<List<TitleValueEntity>> data;

  /**
   * <p>
   * longitude numbers
   * </p>
   * <p>
   * 経線の数量
   * </p>
   * <p>
   * 经线数量
   * </p>
   */
  protected int longitudeNum = DEFAULT_LONGITUDE_NUM;


  /**
   * <p>
   * should display latitude lines?
   * </p>
   * <p>
   * 緯線を表示する?
   * </p>
   * <p>
   * 是否显示纬线
   * </p>
   */
  protected boolean displayLatitude = DEFAULT_DISPLAY_LATITUDE;

  /**
   * <p>
   * latitude numbers
   * </p>
   * <p>
   * 緯線の数量
   * </p>
   * <p>
   * 纬线数量
   * </p>
   */
  protected int latitudeNum = DEFAULT_LATITUDE_NUM;

  /**
   * <p>
   * color for latitude lines
   * </p>
   * <p>
   * 緯線の色
   * </p>
   * <p>
   * 纬线颜色
   * </p>
   */
  protected int latitudeColor = DEFAULT_LATITUDE_COLOR;

  /**
   * <p>
   * color for background
   * </p>
   * <p>
   * 背景の色
   * </p>
   * <p>
   * 蛛网背景色
   * </p>
   */
  protected int backgroundColor = DEFAULT_BACKGROUND_COLOR;

  /*
   * (non-Javadoc)
   *
   * @param context
   *
   * @see cn.limc.ohoscharts.view.AbstractBaseChart#BaseChart(Context)
   */
  public SpiderWebChart(Context context) {
    super(context);
  }

  /*
   * (non-Javadoc)
   *
   * @param context
   *
   * @param attrs
   *
   * @param defStyle
   *
   * @see cn.limc.ohoscharts.view.AbstractBaseChart#BaseChart(Context,AttributeSet,
   * int)
   */
  public SpiderWebChart(Context context, AttrSet attrs, int defStyle) {
    super(context, attrs, defStyle);
  }

  /*
   * (non-Javadoc)
   *
   * @param context
   *
   * @param attrs
   *
   * @see cn.limc.ohoscharts.view.AbstractBaseChart#BaseChart(Context,
   * AttributeSet)
   */
  public SpiderWebChart(Context context, AttrSet attrs) {
    super(context, attrs);
    addDrawTask(this);
  }

  /*
   * (non-Javadoc)
   *
   * <p>Called when is going to draw this chart<p> <p>チャートを書く前、メソッドを呼ぶ<p>
   * <p>绘制图表时调用<p>
   *
   * @param canvas
   *
   * @see ohos.view.View#onDraw(ohos.graphics.Canvas)
   */

  @Override
  public void onDraw(Component component, Canvas canvas) {
    // get safe rect
    int rect = super.getHeight();

    // calculate longitude length
    longitudeLength = (int) ((rect / 2f) * 0.8);

    // calculate position
    position = new Point((int) (super.getWidth() / 2f),
        (int) (super.getHeight() / 2f + 0.2 * longitudeLength));

    drawBorder(canvas);

    // draw this chart
    drawWeb(canvas);

    // draw data on chart
    drawData(canvas);
  }

  /**
   * <p>
   * calculate the points to draw the latitude lines
   * </p>
   * <p>
   * 緯度を使って緯線のポイントを計算する
   * </p>
   * <p>
   * 根据纬度参数的数据计算需要在图上绘制的纬线的点
   * </p>
   *
   * @param pos
   *            <p>
   *            latitude degree
   *            </p>
   *            <p>
   *            緯度
   *            </p>
   *            <p>
   *            纬度
   *            </p>
   *
   * @return　List<PointF>
   *         <p>
   *         result points
   *         </p>
   *         <p>
   *         計算出たポイント
   *         </p>
   *         <p>
   *         计算结果
   *         </p>
   */
  protected List<Point> getWebAxisPoints(float pos) {
    List<Point> points = new ArrayList<Point>();
    for (int i = 0; i < longitudeNum; i++) {
      Point pt = new Point();
      float offsetX = (float) (position.getPointX() - longitudeLength * pos
          * Math.sin(i * 2 * Math.PI / longitudeNum));
      float offsetY = (float) (position.getPointY() - longitudeLength * pos
          * Math.cos(i * 2 * Math.PI / longitudeNum));
      pt.modify(offsetX, offsetY);

      points.add(pt);
    }
    return points;
  }

  /**
   * <p>
   * calculate the points to draw the data
   * </p>
   * <p>
   * データを使ってポイントを計算する
   * </p>
   * <p>
   * 根据参数的数据计算需要在图上绘制的点
   * </p>
   *
   * @param data
   *            <p>
   *            data for calculation
   *            </p>
   *            <p>
   *            計算用データ
   *            </p>
   *            <p>
   *            计算用的数据
   *            </p>
   *
   * @return　List<PointF>
   *         <p>
   *         result points
   *         </p>
   *         <p>
   *         計算出たポイント
   *         </p>
   *         <p>
   *         计算结果
   *         </p>
   */
  protected List<Point> getDataPoints(List<TitleValueEntity> data) {
    List<Point> points = new ArrayList<Point>();
    for (int i = 0; i < longitudeNum; i++) {
      Point pt = new Point();
      float offsetX = (float) (position.getPointX() - data.get(i).getValue() / 10f
          * longitudeLength
          * Math.sin(i * 2 * Math.PI / longitudeNum));
      float offsetY = (float) (position.getPointY() - data.get(i).getValue() / 10f
          * longitudeLength
          * Math.cos(i * 2 * Math.PI / longitudeNum));
      pt.modify(offsetX, offsetY);

      points.add(pt);
    }
    return points;
  }

  /**
   * <p>
   * draw spider web
   * </p>
   * <p>
   * ウェブを書く
   * </p>
   * <p>
   * 绘制蜘蛛网
   * </p>
   *
   * @param canvas
   *            Canvas
   */
  protected void drawWeb(Canvas canvas) {
    Paint mPaintWebFill = new Paint();
    mPaintWebFill.setColor(new Color(Color.GRAY.getValue()));
    mPaintWebFill.setAntiAlias(true);

    Paint mPaintWebBorder = new Paint();
    mPaintWebBorder.setColor(new Color(Color.WHITE.getValue()));
    mPaintWebBorder.setStyle(Paint.Style.STROKE_STYLE);
    mPaintWebBorder.setStrokeWidth(2);
    mPaintWebBorder.setAntiAlias(true);

    Paint mPaintWebInnerBorder = new Paint();
    mPaintWebInnerBorder.setColor(new Color(Color.LTGRAY.getValue()));
    mPaintWebInnerBorder.setStyle(Paint.Style.STROKE_STYLE);
    mPaintWebInnerBorder.setAntiAlias(true);

    Paint mPaintLine = new Paint();
    mPaintLine.setColor(new Color(Color.LTGRAY.getValue()));

    Paint mPaintFont = new Paint();
    mPaintFont.setColor(new Color(Color.LTGRAY.getValue()));

    Path mPath = new Path();
    List<Point> pointList = getWebAxisPoints(1);

    // draw border
    if (null != data) {
      for (int i = 0; i < pointList.size(); i++) {
        Point pt = pointList.get(i);
        if (i == 0) {
          mPath.moveTo(pt.getPointX(), pt.getPointY());
        } else {
          mPath.lineTo(pt.getPointX(), pt.getPointY());
        }

        // draw title
        String title = data.get(0).get(i).getTitle();
        float realx = 0;
        float realy = 0;

        // TODO title position
        if (pt.getPointX() < position.getPointX()) {
          realx = pt.getPointX() - mPaintFont.measureText(title) - 5;
        } else if (pt.getPointX() > position.getPointX()) {
          realx = pt.getPointX() + 5;
        } else {
          realx = pt.getPointX() - mPaintFont.measureText(title) / 2;
        }

        if (pt.getPointY() > position.getPointX()) {
          realy = pt.getPointY() + 10;
        } else if (pt.getPointY() < position.getPointY()) {
          realy = pt.getPointY() - 2;
        } else {
          realy = pt.getPointY() - 5;
        }

        canvas.drawText(mPaintFont, title, realx, realy);
      }
    }
    mPath.close();
    canvas.drawPath(mPath, mPaintWebFill);
    canvas.drawPath(mPath, mPaintWebBorder);

    // draw spider web
    for (int j = 1; j < latitudeNum; j++) {

      Path mPathInner = new Path();
      List<Point> pointListInner = getWebAxisPoints(j * 1f / latitudeNum);

      for (int i = 0; i < pointListInner.size(); i++) {
        Point pt = pointListInner.get(i);
        if (i == 0) {
          mPathInner.moveTo(pt.getPointX(), pt.getPointY());
        } else {
          mPathInner.lineTo(pt.getPointX(), pt.getPointY());
        }
      }
      mPathInner.close();
      canvas.drawPath(mPathInner, mPaintWebInnerBorder);
    }

    // draw longitude lines
    for (int i = 0; i < pointList.size(); i++) {
      Point pt = pointList.get(i);
      canvas.drawLine(position.getPointX(), position.getPointY(), pt.getPointX(), pt.getPointY(),
          mPaintLine);
    }
  }

  /**
   * <p>
   * Draw the data
   * </p>
   * <p>
   * チャートでデータを書く
   * </p>
   * <p>
   * 将数据绘制在图表上
   * </p>
   *
   * @param canvas
   */
  protected void drawData(Canvas canvas) {
    if (null != data) {
      for (int j = 0; j < data.size(); j++) {
        List<TitleValueEntity> list = data.get(j);

        Paint mPaintFill = new Paint();
        mPaintFill.setColor(new Color(COLORS[j]));
        mPaintFill.setStyle(Paint.Style.FILL_STYLE);
        mPaintFill.setAntiAlias(true);
        mPaintFill.setAlpha(0.2f);

        Paint mPaintBorder = new Paint();
        mPaintBorder.setColor(new Color(COLORS[j]));
        mPaintBorder.setStyle(Paint.Style.STROKE_STYLE);
        mPaintBorder.setStrokeWidth(2);
        mPaintBorder.setAntiAlias(true);

        // paint to draw fonts
        Paint mPaintFont = new Paint();
        mPaintFont.setColor(new Color(Color.WHITE.getValue()));

        // paint to draw points
        Paint mPaintPoint = new Paint();
        mPaintPoint.setColor(new Color(COLORS[j]));

        Path mPath = new Path();

        // get points to draw
        List<Point> pointList = getDataPoints(list);
        // initialize path
        for (int i = 0; i < pointList.size(); i++) {
          Point pt = pointList.get(i);
          if (i == 0) {
            mPath.moveTo(pt.getPointX(), pt.getPointY());
          } else {
            mPath.lineTo(pt.getPointX(), pt.getPointY());
          }
          canvas.drawCircle(pt.getPointX(), pt.getPointY(), 3, mPaintPoint);
        }
        mPath.close();

        canvas.drawPath(mPath, mPaintFill);
        canvas.drawPath(mPath, mPaintBorder);
      }
    }
  }

  /**
   * @return the data
   */
  public List<List<TitleValueEntity>> getData() {
    return data;
  }

  /**
   * @param data
   *            the data to set
   */
  public void setData(List<List<TitleValueEntity>> data) {
    this.data = data;
  }

  /**
   * @return the longitudeNum
   */
  public int getLongitudeNum() {
    return longitudeNum;
  }

  /**
   * @param longitudeNum
   *            the longitudeNum to set
   */
  public void setLongitudeNum(int longitudeNum) {
    this.longitudeNum = longitudeNum;
  }

  /**
   * @return the displayLatitude
   */
  public boolean isDisplayLatitude() {
    return displayLatitude;
  }

  /**
   * @param displayLatitude
   *            the displayLatitude to set
   */
  public void setDisplayLatitude(boolean displayLatitude) {
    this.displayLatitude = displayLatitude;
  }

  /**
   * @return the latitudeNum
   */
  public int getLatitudeNum() {
    return latitudeNum;
  }

  /**
   * @param latitudeNum
   *            the latitudeNum to set
   */
  public void setLatitudeNum(int latitudeNum) {
    this.latitudeNum = latitudeNum;
  }

  /**
   * @return the latitudeColor
   */
  public int getLatitudeColor() {
    return latitudeColor;
  }

  /**
   * @param latitudeColor
   *            the latitudeColor to set
   */
  public void setLatitudeColor(int latitudeColor) {
    this.latitudeColor = latitudeColor;
  }
}
